作者:竹茹士弘宜来 | 来源:互联网 | 2023-08-30 23:15
篇首语:本文由编程笔记#小编为大家整理,主要介绍了S5PV210开发板用汇编设置栈和调用C语言相关的知识,希望对你有一定的参考价值。使用C语言前为什么要先用汇编设置栈
篇首语:本文由编程笔记#小编为大家整理,主要介绍了S5PV210开发板用汇编设置栈和调用C语言相关的知识,希望对你有一定的参考价值。
使用C语言前为什么要先用汇编设置栈?
C语言程序运行时需要栈,因为C语言中的局部变量都是用栈来实现的,如果没有设置栈就使用C语言,局部变量就会落空,程序就会死掉,所以在使用C语言前,我们需要先在汇编编写的启动代码中设置栈。
为什么使用51单片机和Ubuntu编写应用程序时我们没有设置栈依然能使用C语言?
原因是在单片机中已经由硬件设置提供了一个默认可用的栈,而在应用程序中我们编写的C程序其实并不是全部,编译器(gcc)在链接时会帮我们自动添加一个头,这个头就是一段引导C程序能够执行的汇编代码,在这个代码中就帮我们的C程序设置了栈及其他的运行时需要。
为什么不同CPU模式下都有各自独立的sp寄存器?
如果各个模式都使用同一个sp,意味着整个程序(操作系统内核程序,用户应用程序)都使用同一个栈,这样做,一旦用户程序编写出错(栈溢出),将会影响操作系统的正常运行,操作系统会崩溃,同时运行在操作系统上的其他应用程序也会崩溃。所以各个模式下要使用不同的sp,使用不同的栈,操作系统有自己的栈,每个应用程序也有自己的栈,互不干扰。
怎么设置栈?
我们要设置栈,并不是设置所有CPU模式下的栈,而是设置当前CPU模式下的栈,因为这个程序可能只用到一种CPU模式。S5PV210复位后默认是进入SVN模式下的,我们设置当前模式下的栈,只需直接操作sp即可。
需要注意的是,栈必须是当前一段可用的内存(可用的意思是这个地方是有被初始化过的可以访问的内存,而且这个内存只会被我们用作栈,不会被其他程序征用)
当前CPU刚复位(刚启动),外部的DDR尚未初始化,需要我们后续对其初始化,所以目前可用的内存只有内部的SRAM(因为它不需初始化即可使用)。因此我们只能在SRAM中找一段内存来作为SVC的栈。
在ARM中,ATPCS(ARM关于程序应该怎么实现的一个规范)要求使用满减栈,结合iROM_application_note中的memory map,可知SVC栈应该设置为0xD0037D80。
示例代码:
start.S
#define WTCON 0xE2700000
#define SVN_STACK 0xD0037D80
.global _start
_start:
ldr r0, =0
ldr r1, =WTCON
str r0, [r1]
ldr sp, =SVN_STACK
mrc p15,0,r0,c1,c0,0;
orr r0, r0, #(1<<12)
mcr p15,0,r0,c1,c0,0;
bl led_blink
b .
led.c
#define GPJ0CON 0xE0200240
#define GPJ0DAT 0xE0200244
#define rGPJ0CON *((volatile unsigned int*)GPJ0CON)
#define rGPJ0DAT *((volatile unsigned int*)GPJ0DAT)
void delay(void);
void led_blink(void)
rGPJ0CON &#61; 0x11111111;
while(1)
rGPJ0DAT &#61; ~(1<<3);
delay();
rGPJ0DAT &#61; ~(1<<4);
delay();
rGPJ0DAT &#61; ~(1<<5);
delay();
void delay(void)
volatile unsigned int i &#61; 3000000;
while(i--);
Makefile
led.bin: start.o led.o
arm-linux-ld -Ttext 0x0 -o led.elf $^
arm-linux-objcopy -O binary led.elf led.bin
arm-linux-objdump -D led.elf > led_elf.dis
gcc mkv210_image.c -o mkx210
./mkx210 led.bin 210.bin
%.o : %.S
arm-linux-gcc -o $&#64; $< -c -nostdlib
%.o : %.c
arm-linux-gcc -o $&#64; $< -c -nostdlib
#这里要加-nostdlib&#xff0c;作用是不使用编译器自带的标准函数库文件
#因为led.c文件中的delay函数与编译器自带的标准函数库中的delay重名
#默认会去使用编译器自带的&#xff0c;我们要使用自己定义的话就会报错
#相当于你一个C程序定义了两个同名函数
clean:
rm *.o *.elf *.bin *.dis mkx210 -f